#### **Procedure:**

This lab contained 3 tasks. Task 1 was to refresh my digital design memory, task 2 was to design and simulate a register file, and task 3 was to design and simulate an ALU. Once the designs were implemented, they were simulated and tested in Modelsim to demonstrate their functionality.

## **Task #2:**

The second task was to implement a register file, the rapid memory unit on a CPU. This was done by first utilizing the register file given in the lab pdf.

Figure 2: The base-level register file provided by the lab pdf

This register file implementation was for a 16x32 register file that was synchronous and had one read port and one write port. The task specified a register file to be made that was also 16x32 with one write port, but with two read ports and asynchronous. To achieve this, the given register file had one more read signal added and the always\_ff block was changed to always\_comb for reading from the file, but the always\_ff block was retained for writing to the file since the specification demanded thus.



Figure 3: This is a block diagram for the overall register file taken from the lab specification

# **Task #3:**

The third task was to implement an ALU unit from a CPU. This was done by first taking the specifications required by the lab and mapping them out to create a block diagram.



Figure 4: The overall block diagram for the ALU

A and B are fed into fulladders, and and or modules bit-by-bit. The outputs from each are then passed through to a 3x1 MUX and processed to output result. Outputs from the fullAdder are used to determine the carrier inputs in a feedback loop. The ALUControl signal is used as a mux

control signal and then the Result combined with the carrier inputs is used to determine the ALUFlags signal.

Following the block diagram, the circuit was implemented in Quartus using System Verilog. It was then tested in Modelsim by varying a ,b, ALUControl, to see the change in Result and ALUFlags. Then a, b and ALUControl were varied using testvectors, see attached alu.tv screenshot in Appendix for details.

| Test                         | ALUControl[1:0 | A            | В            | Y            | ALUFlag<br>s |
|------------------------------|----------------|--------------|--------------|--------------|--------------|
| ADD 0+0                      | 0              | 0000000      | 0000000      | 0000000      | 4            |
| ADD 0+(-1)                   | 0              | 0000000      | FFFFFFF<br>F | FFFFFFF      | 8            |
| ADD 1+(-1)                   | 0              | 0000000      | FFFFFFF      | 0000000      | 6            |
| ADD FF+1                     | 0              | 000000F<br>F | 0000000      |              | 0            |
| SUB 0-0                      | 1              | 0000000      | 0000000      | 0000000      | 6            |
| SUB 0-(-1)                   | 1              | 0000000      | FFFFFFF      | 0000000      | 0            |
| SUB 1-1                      | 1              | 0000000      | 0000001      | 00000000     | 6            |
| SUB 100-1                    | 1              | 0000010      | 00000001     | 000000FF     | 2            |
| AND<br>FFFFFFFF,<br>FFFFFFFF | 2              | FFFFFFF<br>F | FFFFFFF      | FFFFFFF      | 10           |
| AND<br>FFFFFFFF,<br>12345678 | 2              | FFFFFFF<br>F | 1234567<br>8 | 1234567<br>8 | <b>a</b> 2   |
| AND<br>12345678,<br>87654321 | 2              | 1234567<br>8 | 87654321     | 02244220     | 0            |
| AND<br>00000000,<br>FFFFFFF  | 2              | 0000000      | FFFFFFF      | 00000000     | 4            |
| OR FFFFFFFF<br>, FFFFFFFF    | 3              | FFFFFFF<br>F | FFFFFFF      | FFFFFFF      | 10           |
| OR 12345678<br>, 87654321    | 3              | 1234567<br>8 | 87654321     | 97755779     | 9            |
| OR 00000000<br>, FFFFFFFF    | 3              | 0000000      | FFFFFFF      | FFFFFFF      | 8            |
| OR 00000000<br>, 00000000    | 3              | 0000000      | 00000000     | 00000000     | 6            |

Figure 5: The values that were varied for the test vector used and the resulting values

I used this table to verify that all four ALU operations work as intended and filled in the missing values.

### **Results**

**Task #2:** Register File

After implementing the register file in Quartus, I ran ModelSim to test it.



Figure 6: ModelSim waves for testing writes delays (1 cycle)



Figure 7: ModelSim waves for testing read delays (0 cycles)



Figure 8: ModelSim waves for testing read update delays after a write (1 cycle after the write)

The three functionalities that are being tested are given by the lab specification:

- 1. Write data is written into the register file the clock cycle **after** wr\_en is asserted.
- 2. Read data is updated to the register data at an address the **same** cycle the address was provided. Do this for both read addresses and data outputs.
- 3. Read data is updated to write data at an address the cycle **after** the address was provided if the write address is the same and wr\_en asserted. Do this for both read addresses and data outputs.

As seen in the waveforms above, data is written only 1 cycle after 'wr\_en' is equal to 1. Read data pushes through in the very same cycle since it is done using combinational logic. Read data is updated 1 cycle after if a write is performed at the same address as the read.

#### Task #3: ALU

After implementing the ALU in Quartus, I ran ModelSim to test it.



Figure 9: Waveform generated by running a test vector through the ALU

As seen from the figure, Result is the output of performing the function defined by ALUControl and the ALUFlags are updated based on the carriers within the ALU and the Result value.

# **Appendix**

See the following list for the order:

Task 2: reg\_file.sv

reg\_file\_testbench.sv

Task 3: alu.sv

alu\_testbench.sv

singleALU.sv

fullAdder.sv

alu.tv

See the attached documents for the code:

Date: April 07, 2023

```
// Eugene Ngo
      // 4/7/2023
 3
      // CSE 469
 4
      // Lab 1 Task 2
      // This is the top level module // This register file is 16x32, has two read ports, one write port, and is asynchronous
      // as specified
 8
 9
     module reg_file (input logic clk, wr_en, input logic [31:0] write_data,
                           input logic [3:0] write_addr,
input logic [3:0] read_addr1, read_addr2,
10
11
12
                           output logic [31:0] read_data1, read_data2);
13
14
         logic [15:0][31:0] memory;
15
16
         always_ff @ (posedge clk) begin
17
             if (wr_en) begin
18
                memory[write_addr] <= write_data;</pre>
19
20
21
             end
         end
22
         always_comb begin
23
             read_data1 = memory[read_addr1];
24
25
             read_data2 = memory[read_addr2];
26
27
      endmodule
28
29
      // reg_file__testbench tests all expected, unexpected, and edgecase behaviors
30
      module reg_file_testbench();
         logic clk, wr_en;
logic [31:0] write_data, read_data1, read_data2;
31
32
33
         logic [3:0] write_addr, read_addr1, read_addr2;
34
35
         reg_file dut (.clk, .wr_en, .write_data, .write_addr, .read_addr1, .read_addr2, .
      read_data1, .read_data2);
36
37
         parameter clock_period = 10000;
38
39
         integer i;
40
41
         initial begin // Set up the clock
42
43
             for (i=0; i<1000; i++) begin: clockCount</pre>
44
45
46
                forever #(clock_period /2) clk <= ~clk;</pre>
             end
47
48
49
         end
         initial begin
50
51
52
53
54
55
56
             $display("%t Behavior check", $time);
            @(posedge clk);
             write_data = 32'b0;
                                                @(posedge clk);
             write\_addr = 4'b0010;
                                                @(posedge clk);
57
                                                @(posedge clk);
58
59
             write_data = 32'b1;
                                                @(posedge clk);
60
             write\_addr = 4'b0011;
                                                @(posedge clk);
61
                                                @(posedge clk);
             // Testing functionality of
// same cycle reads.
read_addr1 = 4'b0010;  @(po
read_addr2 = 4'b0011;  @(po
62
63
64
                                                @(posedge clk);
65
                                                @(posedge clk);
66
                    // Testing the functionality of
// 1 cycle delayed reads
67
68
            // after an updated write write_addr = 4'b0010; @(
69
70
                                                @(posedge clk);
             read\_addr1 = 4'b0010;
71
                                                @(posedge clk);
72
                                                @(posedge clk);
```

90

91

endmodule

Project: reg\_file

68

```
// Eugene Ngo
      // 4/7/2023
 3
      // CSE 469
 4
      // Lab 1 Task 2
      // reg_file__testbench tests all expected, unexpected, and edgecase behaviors
module reg_file_testbench();
  logic clk, wr_en;
  logic [31:0] write_data, read_data1, read_data2;
  logic [3:0] write_addr, read_addr1, read_addr2;
 7
 8
 9
10
11
          reg_file dut (.clk, .wr_en, .write_data, .write_addr, .read_addr1, .read_addr2, .
12
      read_data1, .read_data2);
13
14
          parameter clock_period = 10000;
15
16
          integer i;
17
18
19
          initial begin // Set up the clock
             clk <= 0;
             for (i=0; i<1000; i++) begin: clockCount
  forever #(clock_period /2) clk <= ~clk;</pre>
20
21
22
             end
end
          initial begin
             $display("%t Behavior check", $time);
                     // Testing functionality of
             @(posedge clk);
             write_data = 32'b0;
                                                  @(posedge clk);
             write_addr = 4'b0010;
                                                  @(posedge clk);
                                                  @(posedge clk);
             write_data = 32'b1;
                                                  @(posedge clk);
             write_addr = 4'b0011;
                                                  @(posedge clk);
                                                  @(posedge clk);
                     // Testing functionality of
             // same cycle reads.
read_addr1 = 4'b0010;
                                                  @(posedge clk);
             read_addr2 = 4'b0011;
                                                  @(posedge clk);
                     // Testing the functionality of
             // 1 cycle delayed reads
// after an updated write
write_addr = 4'b0010; @(
                                                  @(posedge clk);
             read\_addr1 = 4'b0010;
                                                  @(posedge clk);
                                                  @(posedge clk);
             // read_data1 should update to 1 now.
             write_data = 32'b0;
                                                  @(posedge clk);
             write_addr = 4'b00\dot{1}1;
                                                  @(posedge clk);
             read\_addr2 = 4'b0011;
                                                  @(posedge clk);
                                                  @(posedge clk);
             // read_data2 should update to 0 now.
58
59
60
61
62
          end
63
64
65
66
      endmodule
67
```

```
// Eugene Ngo
            // 4/7/2023
  3
            // CSE 469
  4
            // Lab 1 Task 3
           8
  9
10
11
12
13
                    // 00 = add
14
                    // 01 = subtract
15
                    // 10 = AND
16
                    // 11 = OR
17
18
                    logic [31:0] carries;
19
20
21
                    singleALU setCarries (.a(a[0]), .b(b[0]), .carryIn(ALUControl[0]),
                                                                           .ALUControl(ALUControl), .Result(Result[0]),
22
                                                                           .carryOut(carries[0]));
23
24
25
26
27
                    genvar i;
                    generate
                           for (i = 1; i < 32; i++) begin: ALUPipeline
                                   singleALU results (.a(a[i]), .b(b[i]), .carryIn(carries[i - 1]),
28
29
                                                                                .ALUControl(ALUControl), .Result(Result[i]),
                                                                                .carryOut(carries[i]));
30
31
32
33
34
35
36
37
                           end // end loop
                    endgenerate // end generate
                    // Setting flags:
                    xor overFlowCheck (ALUFlags[0], carries[31], carries[30]);
                    assign ALUFlags[1] = carries[31];
                    // Inefficient and bad style. RTL would be better.
38
                    nor zeroChecker
                           (ALUFlags[2], Result[31], Result[30], Result[29], Result[28], Result[27], Result[26], Result[25], Result[24], Result[23], Result[22], Result[21], Result[20], Result[19], Result[18], Result[17], Result[16], Result[15], Result[14], Result[13], Result[17], Result[16], Resu
39
40
41
42
43
                             Result[12], Result[11], Result[10], Result[9], Result[8],
44
                             Result[7], Result[6], Result[5], Result[4], Result[3],
Result[2], Result[1], Result[0]);
45
46
47
                    assign ALUFlags[3] = Result[31];
48
49
50
51
52
53
54
55
56
57
            endmodule
            // alu_testbench tests all expected, unexpected, and edgecase behaviors
            module alu_testbench();
  logic [31:0] a,b;
  logic [1:0] ALUControl;
  logic [31:0] Result;
                    logic [3:0] ALUFlags;
                    logic clk;
58
                    logic [103:0] testvectors [1000:0];
59
                    alu dut (.a(a), .b(b), .ALUControl(ALUControl), .Result(Result),
60
61
                                           .ALUFlags(ALUFlags));
62
63
                    parameter CLOCK_PERIOD = 100;
64
65
                    initial clk = 1;
66
67
                    always begin
68
                                  #(CLOCK_PERIOD/2);
69
                                  c1k = \sim c1k;
70
                    end
71
72
                    initial begin
                           $readmemh("alu.tv", testvectors);
73
```

```
// Eugene Ngo
      // 4/7/2023
     // CSE 469
// Lab 1 Task 3
 3
 4
     // alu_testbench tests all expected, unexpected, and edgecase behaviors
module alu_testbench();
  logic [31:0] a,b;
  logic [1:0] ALUControl;
  logic [31:0] Result;
  logic [3:0] ALUFlags;
  logic clk;
 6
7
 8
 9
10
11
         logic clk;
logic [103:0] testvectors [1000:0];
12
13
14
         15
16
17
18
19
20
21
22
         parameter CLOCK_PERIOD = 100;
         initial clk = 1;
         always begin
23
24
25
26
27
28
29
30
31
                #(CLOCK_PERIOD/2);
                clk = \sim clk;
         end
         initial begin
             $readmemh("alu.tv", testvectors);
     33
34
35
```

```
// Eugene Ngo
      // 4/7/2023
 3
      // CSE 469
 4
      // Lab 1 Task 3
      // singleALU implements ALU logic for single bits which can be combined to // make up large ALUs.
 7
      module singleALU (a, b, carryIn, ALUControl, Result, carryOut);
  input logic a, b, carryIn;
  input logic [1:0] ALUControl;
 8
 9
10
11
          output logic Result, carryOut;
12
13
14
          logic [2:0] outputs;
15
16
17
          and andValue (outputs[1], b, a);
or orValue (outputs[2], b, a);
18
19
20
21
22
          // Selecting B (Addition = A+B+0, Subtraction = A+(~B)+1)
// logic subtractSelector = ALUControl[0];
          // MUX to select B (ALUControl[0] == 1 = select ~B)
23
24
25
26
27
28
29
30
31
32
33
34
35
37
          wire middlevalues[1:0];
          and selectB (middleValues[0], ~ALUControl[0], b);
and selectNotB (middleValues[1], ALUControl[0], ~b);
          logic selectedB;
          or selectedBvalue (selectedB, middlevalues[0], middlevalues[1]);
          fullAdder fullAddedValue (.a(a), .b(selectedB), .carryIn(carryIn)
                                              .Result(outputs[0]), .carryOut(carryOut));
          // MUX to select between computed values (add, sub, and, or)
          always_comb begin
              case (ALUControl)
  2'b00: Result = outputs[0];
                   2'b01: Result = outputs[0];
2'b10: Result = outputs[1];
38
39
40
41
42
                   2'b11: Result_= outputs[2];
                  default: Result = 1'bX;
              endcase // end case statements
43
          end // end comb block
44
          // End of module
45
46
      endmodule
47
```

Page 1 of 1

```
// Eugene Ngo
// 4/7/2023
// CSE 469
// Lab 1 Task 3
// A module to implement a fullAdder.
// This file was reused from a given file in EE 371
module fullAdder (a, b, carryIn, Result, carryOut);

input logic a, b, carryIn;
output logic Result, carryOut;

assign Result = a ^ b ^ carryIn;
assign carryOut = (a & b) | (carryIn & (a ^ b));

endmodule
```

 $0\_00000000\_000000000\_000000000\_4$ 2 3  $0\_00000000\_FFFFFFFFFFFFF8$ 4 5 6 7 1\_00000100\_00000001\_000000FF\_2 8 2\_FFFFFFFF\_12345678\_12345678\_2 9 10 2\_12345678\_87654321\_02244220\_0 11 12 2\_00000000\_ffffffff\_00000000\_4 3\_FFFFFFF\_FFFFFFFF\_8 3\_12345678\_87654321\_97755779\_9 13 14 15 3\_0000000\_FFFFFFFF\_FFFFF\_8 16 3\_00000000\_00000000\_00000000\_6